1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17 package org.orekit.propagation.semianalytical.dsst.forces;
18
19 import java.io.NotSerializableException;
20 import java.io.Serializable;
21 import java.util.ArrayList;
22 import java.util.HashMap;
23 import java.util.List;
24 import java.util.Map;
25 import java.util.Set;
26 import java.util.SortedSet;
27
28 import org.hipparchus.analysis.UnivariateVectorFunction;
29 import org.hipparchus.geometry.euclidean.threed.Vector3D;
30 import org.hipparchus.util.FastMath;
31 import org.orekit.attitudes.Attitude;
32 import org.orekit.attitudes.AttitudeProvider;
33 import org.orekit.errors.OrekitException;
34 import org.orekit.errors.OrekitExceptionWrapper;
35 import org.orekit.forces.ForceModel;
36 import org.orekit.frames.Frame;
37 import org.orekit.orbits.EquinoctialOrbit;
38 import org.orekit.orbits.Orbit;
39 import org.orekit.orbits.PositionAngle;
40 import org.orekit.propagation.SpacecraftState;
41 import org.orekit.propagation.numerical.TimeDerivativesEquations;
42 import org.orekit.propagation.semianalytical.dsst.utilities.AuxiliaryElements;
43 import org.orekit.propagation.semianalytical.dsst.utilities.CjSjCoefficient;
44 import org.orekit.propagation.semianalytical.dsst.utilities.ShortPeriodicsInterpolatedCoefficient;
45 import org.orekit.time.AbsoluteDate;
46 import org.orekit.utils.TimeSpanMap;
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74 public abstract class AbstractGaussianContribution implements DSSTForceModel {
75
76
77 private static final int[] GAUSS_ORDER = {12, 16, 20, 24, 32, 40, 48};
78
79
80 private static final int MAX_ORDER_RANK = GAUSS_ORDER.length - 1;
81
82
83 private static final int INTERPOLATION_POINTS = 3;
84
85
86 private static final int JMAX = 12;
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101 private static final int I = 1;
102
103
104
105
106 protected double a;
107
108 protected double k;
109
110 protected double h;
111
112 protected double q;
113
114 protected double p;
115
116
117 protected double ecc;
118
119
120 protected double n;
121
122
123 protected double lm;
124
125
126 protected Vector3D f;
127
128 protected Vector3D g;
129
130 protected Vector3D w;
131
132
133 protected double A;
134
135 protected double B;
136
137 protected double C;
138
139
140 protected double ton2a;
141
142 protected double ooA;
143
144 protected double ooAB;
145
146 protected double co2AB;
147
148 protected double ooBpo;
149
150 protected double ooMu;
151
152 protected double mu;
153
154
155
156
157 private final ForceModel contribution;
158
159
160 private final double threshold;
161
162
163 private GaussQuadrature integrator;
164
165
166 private boolean isDirty;
167
168
169 private AttitudeProvider attitudeProvider;
170
171
172 private final String coefficientsKeyPrefix;
173
174
175 private GaussianShortPeriodicCoefficients gaussianSPCoefs;
176
177
178
179
180
181
182 protected AbstractGaussianContribution(final String coefficientsKeyPrefix,
183 final double threshold,
184 final ForceModel contribution) {
185 this.coefficientsKeyPrefix = coefficientsKeyPrefix;
186 this.contribution = contribution;
187 this.threshold = threshold;
188 this.integrator = new GaussQuadrature(GAUSS_ORDER[MAX_ORDER_RANK]);
189 this.isDirty = true;
190 }
191
192
193 @Override
194 public List<ShortPeriodTerms> initialize(final AuxiliaryElements aux, final boolean meanOnly) {
195
196 final List<ShortPeriodTerms> list = new ArrayList<ShortPeriodTerms>();
197 gaussianSPCoefs = new GaussianShortPeriodicCoefficients(coefficientsKeyPrefix,
198 JMAX, INTERPOLATION_POINTS,
199 new TimeSpanMap<Slot>(new Slot(JMAX, INTERPOLATION_POINTS)));
200 list.add(gaussianSPCoefs);
201 return list;
202
203 }
204
205
206 @Override
207 public void initializeStep(final AuxiliaryElements aux)
208 throws OrekitException {
209
210
211 a = aux.getSma();
212 k = aux.getK();
213 h = aux.getH();
214 q = aux.getQ();
215 p = aux.getP();
216
217
218 ecc = aux.getEcc();
219
220
221 A = aux.getA();
222 B = aux.getB();
223 C = aux.getC();
224
225
226 f = aux.getVectorF();
227 g = aux.getVectorG();
228 w = aux.getVectorW();
229
230
231 n = aux.getMeanMotion();
232
233
234 lm = aux.getLM();
235
236
237 ooA = 1. / A;
238
239 ooAB = ooA / B;
240
241 co2AB = C * ooAB / 2.;
242
243 ooBpo = 1. / (1. + B);
244
245 ton2a = 2. / (n * n * a);
246
247 mu = aux.getMu();
248
249 ooMu = 1. / mu;
250 }
251
252
253 @Override
254 public double[] getMeanElementRate(final SpacecraftState state) throws OrekitException {
255
256 double[] meanElementRate = new double[6];
257
258 final double[] ll = getLLimits(state);
259
260 if (ll[0] < ll[1]) {
261 meanElementRate = getMeanElementRate(state, integrator, ll[0], ll[1]);
262 if (isDirty) {
263 boolean next = true;
264 for (int i = 0; i < MAX_ORDER_RANK && next; i++) {
265 final double[] meanRates = getMeanElementRate(state, new GaussQuadrature(GAUSS_ORDER[i]), ll[0], ll[1]);
266 if (getRatesDiff(meanElementRate, meanRates) < threshold) {
267 integrator = new GaussQuadrature(GAUSS_ORDER[i]);
268 next = false;
269 }
270 }
271 isDirty = false;
272 }
273 }
274 return meanElementRate;
275 }
276
277
278
279
280
281
282
283 protected Vector3D getAcceleration(final SpacecraftState state)
284 throws OrekitException {
285 final AccelerationRetriever retriever = new AccelerationRetriever(state);
286 contribution.addContribution(state, retriever);
287
288 return retriever.getAcceleration();
289 }
290
291
292
293
294
295
296
297 protected abstract double[] getLLimits(final SpacecraftState state) throws OrekitException;
298
299
300
301
302
303
304
305
306
307
308 private double[] getMeanElementRate(final SpacecraftState state,
309 final GaussQuadrature gauss,
310 final double low,
311 final double high) throws OrekitException {
312 final double[] meanElementRate = gauss.integrate(new IntegrableFunction(state, true, 0), low, high);
313
314 final double coef = 1. / (2. * FastMath.PI * B);
315
316 for (int i = 0; i < 6; i++) {
317 meanElementRate[i] *= coef;
318 }
319 return meanElementRate;
320 }
321
322
323
324
325
326
327
328 private double getRatesDiff(final double[] meanRef, final double[] meanCur) {
329 double maxDiff = FastMath.abs(meanRef[0] - meanCur[0]) / a;
330
331 for (int i = 1; i < meanRef.length; i++) {
332 final double diff = FastMath.abs(meanRef[i] - meanCur[i]);
333 if (maxDiff < diff) maxDiff = diff;
334 }
335 return maxDiff;
336 }
337
338
339 @Override
340 public void registerAttitudeProvider(final AttitudeProvider provider) {
341 this.attitudeProvider = provider;
342 }
343
344
345 @Override
346 public void updateShortPeriodTerms(final SpacecraftState ... meanStates)
347 throws OrekitException {
348
349 final Slot slot = gaussianSPCoefs.createSlot(meanStates);
350 for (final SpacecraftState meanState : meanStates) {
351 initializeStep(new AuxiliaryElements(meanState.getOrbit(), I));
352 final double[][] currentRhoSigmaj = computeRhoSigmaCoefficients(meanState.getDate());
353 final FourierCjSjCoefficients fourierCjSj = new FourierCjSjCoefficients(meanState, JMAX);
354 final UijVijCoefficients uijvij = new UijVijCoefficients(currentRhoSigmaj, fourierCjSj, JMAX);
355 gaussianSPCoefs.computeCoefficients(meanState, slot, fourierCjSj, uijvij, n, a);
356 }
357
358 }
359
360
361
362
363
364
365
366
367
368
369
370 private double[][] computeRhoSigmaCoefficients(final AbsoluteDate date) {
371 final double[][] currentRhoSigmaj = new double[2][3 * JMAX + 1];
372 final CjSjCoefficient cjsjKH = new CjSjCoefficient(k, h);
373 final double b = 1. / (1 + B);
374
375
376 double mbtj = 1;
377
378 for (int j = 1; j <= 3 * JMAX; j++) {
379
380
381 mbtj *= -b;
382 final double coef = (1 + j * B) * mbtj;
383 currentRhoSigmaj[0][j] = coef * cjsjKH.getCj(j);
384 currentRhoSigmaj[1][j] = coef * cjsjKH.getSj(j);
385 }
386 return currentRhoSigmaj;
387 }
388
389
390 private static class AccelerationRetriever implements TimeDerivativesEquations {
391
392
393 private Vector3D acceleration;
394
395
396 private final SpacecraftState state;
397
398
399
400
401 AccelerationRetriever(final SpacecraftState state) {
402 this.acceleration = Vector3D.ZERO;
403 this.state = state;
404 }
405
406
407 @Override
408 public void addKeplerContribution(final double mu) {
409 }
410
411
412 @Override
413 public void addXYZAcceleration(final double x, final double y, final double z) {
414 acceleration = new Vector3D(x, y, z);
415 }
416
417
418 @Override
419 public void addAcceleration(final Vector3D gamma, final Frame frame)
420 throws OrekitException {
421 acceleration = frame.getTransformTo(state.getFrame(),
422 state.getDate()).transformVector(gamma);
423 }
424
425
426 @Override
427 public void addMassDerivative(final double q) {
428 }
429
430
431
432
433 public Vector3D getAcceleration() {
434 return acceleration;
435 }
436
437 }
438
439
440 private class IntegrableFunction implements UnivariateVectorFunction {
441
442
443 private final SpacecraftState state;
444
445
446
447 private final boolean meanMode;
448
449
450
451
452
453 private final int j;
454
455
456
457
458
459
460
461 IntegrableFunction(final SpacecraftState state, final boolean meanMode, final int j) {
462 this.state = state;
463 this.meanMode = meanMode;
464 this.j = j;
465 }
466
467
468 @Override
469 public double[] value(final double x) {
470
471
472 final double shiftedLm = trueToMean(x);
473 final double dLm = shiftedLm - lm;
474 final double dt = dLm / n;
475
476 final double cosL = FastMath.cos(x);
477 final double sinL = FastMath.sin(x);
478 final double roa = B * B / (1. + h * sinL + k * cosL);
479 final double roa2 = roa * roa;
480 final double r = a * roa;
481 final double X = r * cosL;
482 final double Y = r * sinL;
483 final double naob = n * a / B;
484 final double Xdot = -naob * (h + sinL);
485 final double Ydot = naob * (k + cosL);
486 final Vector3D vel = new Vector3D(Xdot, f, Ydot, g);
487
488
489 Vector3D acc = Vector3D.ZERO;
490 try {
491
492
493 final Orbit shiftedOrbit = state.getOrbit().shiftedBy(dt);
494
495
496 final Orbit recomposedOrbit =
497 new EquinoctialOrbit(shiftedOrbit.getA(),
498 shiftedOrbit.getEquinoctialEx(),
499 shiftedOrbit.getEquinoctialEy(),
500 shiftedOrbit.getHx(),
501 shiftedOrbit.getHy(),
502 shiftedOrbit.getLv(),
503 PositionAngle.TRUE,
504 shiftedOrbit.getFrame(),
505 state.getDate(),
506 shiftedOrbit.getMu());
507
508
509 final Attitude recomposedAttitude =
510 attitudeProvider.getAttitude(recomposedOrbit,
511 recomposedOrbit.getDate(),
512 recomposedOrbit.getFrame());
513
514
515 final SpacecraftState shiftedState =
516 new SpacecraftState(recomposedOrbit, recomposedAttitude, state.getMass());
517
518 acc = getAcceleration(shiftedState);
519
520 } catch (OrekitException oe) {
521 throw new OrekitExceptionWrapper(oe);
522 }
523
524 final double[] deriv = new double[6];
525
526 deriv[0] = getAoV(vel).dotProduct(acc);
527
528 deriv[1] = getKoV(X, Y, Xdot, Ydot).dotProduct(acc);
529
530 deriv[2] = getHoV(X, Y, Xdot, Ydot).dotProduct(acc);
531
532 deriv[3] = getQoV(X).dotProduct(acc);
533
534 deriv[4] = getPoV(Y).dotProduct(acc);
535
536 deriv[5] = getLoV(X, Y, Xdot, Ydot).dotProduct(acc);
537
538
539 double[] val = null;
540 if (meanMode) {
541 val = new double[6];
542 for (int i = 0; i < 6; i++) {
543
544 val[i] = roa2 * deriv[i];
545 }
546 } else {
547 val = new double[12];
548
549 final double cosjL = j == 1 ? cosL : FastMath.cos(j * x);
550 final double sinjL = j == 1 ? sinL : FastMath.sin(j * x);
551
552 for (int i = 0; i < 6; i++) {
553
554 val[i] = cosjL * deriv[i];
555
556 val[i + 6] = sinjL * deriv[i];
557 }
558 }
559 return val;
560 }
561
562
563
564
565
566 private double trueToEccentric (final double lv) {
567 final double cosLv = FastMath.cos(lv);
568 final double sinLv = FastMath.sin(lv);
569 final double num = h * cosLv - k * sinLv;
570 final double den = B + 1 + k * cosLv + h * sinLv;
571 return lv + 2 * FastMath.atan(num / den);
572 }
573
574
575
576
577
578 private double eccentricToMean (final double le) {
579 return le - k * FastMath.sin(le) + h * FastMath.cos(le);
580 }
581
582
583
584
585
586 private double trueToMean (final double lv) {
587 return eccentricToMean(trueToEccentric(lv));
588 }
589
590
591
592
593
594 private Vector3D getAoV(final Vector3D vel) {
595 return new Vector3D(ton2a, vel);
596 }
597
598
599
600
601
602
603
604
605 private Vector3D getHoV(final double X, final double Y, final double Xdot, final double Ydot) {
606 final double kf = (2. * Xdot * Y - X * Ydot) * ooMu;
607 final double kg = X * Xdot * ooMu;
608 final double kw = k * (I * q * Y - p * X) * ooAB;
609 return new Vector3D(kf, f, -kg, g, kw, w);
610 }
611
612
613
614
615
616
617
618
619 private Vector3D getKoV(final double X, final double Y, final double Xdot, final double Ydot) {
620 final double kf = Y * Ydot * ooMu;
621 final double kg = (2. * X * Ydot - Xdot * Y) * ooMu;
622 final double kw = h * (I * q * Y - p * X) * ooAB;
623 return new Vector3D(-kf, f, kg, g, -kw, w);
624 }
625
626
627
628
629
630 private Vector3D getPoV(final double Y) {
631 return new Vector3D(co2AB * Y, w);
632 }
633
634
635
636
637
638 private Vector3D getQoV(final double X) {
639 return new Vector3D(I * co2AB * X, w);
640 }
641
642
643
644
645
646
647
648
649 private Vector3D getLoV(final double X, final double Y, final double Xdot, final double Ydot) {
650 final Vector3D pos = new Vector3D(X, f, Y, g);
651 final Vector3D v2 = new Vector3D(k, getHoV(X, Y, Xdot, Ydot), -h, getKoV(X, Y, Xdot, Ydot));
652 return new Vector3D(-2. * ooA, pos, ooBpo, v2, (I * q * Y - p * X) * ooA, w);
653 }
654
655 }
656
657
658
659
660
661 private static class GaussQuadrature {
662
663
664
665
666
667
668 private static final double[] P_12 = {
669 -0.98156063424671910000,
670 -0.90411725637047490000,
671 -0.76990267419430470000,
672 -0.58731795428661740000,
673 -0.36783149899818024000,
674 -0.12523340851146890000,
675 0.12523340851146890000,
676 0.36783149899818024000,
677 0.58731795428661740000,
678 0.76990267419430470000,
679 0.90411725637047490000,
680 0.98156063424671910000
681 };
682
683
684 private static final double[] W_12 = {
685 0.04717533638651220000,
686 0.10693932599531830000,
687 0.16007832854334633000,
688 0.20316742672306584000,
689 0.23349253653835478000,
690 0.24914704581340286000,
691 0.24914704581340286000,
692 0.23349253653835478000,
693 0.20316742672306584000,
694 0.16007832854334633000,
695 0.10693932599531830000,
696 0.04717533638651220000
697 };
698
699
700 private static final double[] P_16 = {
701 -0.98940093499164990000,
702 -0.94457502307323260000,
703 -0.86563120238783160000,
704 -0.75540440835500310000,
705 -0.61787624440264380000,
706 -0.45801677765722737000,
707 -0.28160355077925890000,
708 -0.09501250983763745000,
709 0.09501250983763745000,
710 0.28160355077925890000,
711 0.45801677765722737000,
712 0.61787624440264380000,
713 0.75540440835500310000,
714 0.86563120238783160000,
715 0.94457502307323260000,
716 0.98940093499164990000
717 };
718
719
720 private static final double[] W_16 = {
721 0.02715245941175405800,
722 0.06225352393864777000,
723 0.09515851168249283000,
724 0.12462897125553388000,
725 0.14959598881657685000,
726 0.16915651939500256000,
727 0.18260341504492360000,
728 0.18945061045506847000,
729 0.18945061045506847000,
730 0.18260341504492360000,
731 0.16915651939500256000,
732 0.14959598881657685000,
733 0.12462897125553388000,
734 0.09515851168249283000,
735 0.06225352393864777000,
736 0.02715245941175405800
737 };
738
739
740 private static final double[] P_20 = {
741 -0.99312859918509490000,
742 -0.96397192727791390000,
743 -0.91223442825132600000,
744 -0.83911697182221890000,
745 -0.74633190646015080000,
746 -0.63605368072651510000,
747 -0.51086700195082700000,
748 -0.37370608871541955000,
749 -0.22778585114164507000,
750 -0.07652652113349734000,
751 0.07652652113349734000,
752 0.22778585114164507000,
753 0.37370608871541955000,
754 0.51086700195082700000,
755 0.63605368072651510000,
756 0.74633190646015080000,
757 0.83911697182221890000,
758 0.91223442825132600000,
759 0.96397192727791390000,
760 0.99312859918509490000
761 };
762
763
764 private static final double[] W_20 = {
765 0.01761400713915226400,
766 0.04060142980038684000,
767 0.06267204833410904000,
768 0.08327674157670477000,
769 0.10193011981724048000,
770 0.11819453196151844000,
771 0.13168863844917678000,
772 0.14209610931838212000,
773 0.14917298647260380000,
774 0.15275338713072600000,
775 0.15275338713072600000,
776 0.14917298647260380000,
777 0.14209610931838212000,
778 0.13168863844917678000,
779 0.11819453196151844000,
780 0.10193011981724048000,
781 0.08327674157670477000,
782 0.06267204833410904000,
783 0.04060142980038684000,
784 0.01761400713915226400
785 };
786
787
788 private static final double[] P_24 = {
789 -0.99518721999702130000,
790 -0.97472855597130950000,
791 -0.93827455200273270000,
792 -0.88641552700440100000,
793 -0.82000198597390300000,
794 -0.74012419157855440000,
795 -0.64809365193697550000,
796 -0.54542147138883950000,
797 -0.43379350762604520000,
798 -0.31504267969616340000,
799 -0.19111886747361634000,
800 -0.06405689286260563000,
801 0.06405689286260563000,
802 0.19111886747361634000,
803 0.31504267969616340000,
804 0.43379350762604520000,
805 0.54542147138883950000,
806 0.64809365193697550000,
807 0.74012419157855440000,
808 0.82000198597390300000,
809 0.88641552700440100000,
810 0.93827455200273270000,
811 0.97472855597130950000,
812 0.99518721999702130000
813 };
814
815
816 private static final double[] W_24 = {
817 0.01234122979998733500,
818 0.02853138862893380600,
819 0.04427743881741981000,
820 0.05929858491543691500,
821 0.07334648141108027000,
822 0.08619016153195320000,
823 0.09761865210411391000,
824 0.10744427011596558000,
825 0.11550566805372553000,
826 0.12167047292780335000,
827 0.12583745634682825000,
828 0.12793819534675221000,
829 0.12793819534675221000,
830 0.12583745634682825000,
831 0.12167047292780335000,
832 0.11550566805372553000,
833 0.10744427011596558000,
834 0.09761865210411391000,
835 0.08619016153195320000,
836 0.07334648141108027000,
837 0.05929858491543691500,
838 0.04427743881741981000,
839 0.02853138862893380600,
840 0.01234122979998733500
841 };
842
843
844 private static final double[] P_32 = {
845 -0.99726386184948160000,
846 -0.98561151154526840000,
847 -0.96476225558750640000,
848 -0.93490607593773970000,
849 -0.89632115576605220000,
850 -0.84936761373256990000,
851 -0.79448379596794250000,
852 -0.73218211874028970000,
853 -0.66304426693021520000,
854 -0.58771575724076230000,
855 -0.50689990893222950000,
856 -0.42135127613063540000,
857 -0.33186860228212767000,
858 -0.23928736225213710000,
859 -0.14447196158279646000,
860 -0.04830766568773831000,
861 0.04830766568773831000,
862 0.14447196158279646000,
863 0.23928736225213710000,
864 0.33186860228212767000,
865 0.42135127613063540000,
866 0.50689990893222950000,
867 0.58771575724076230000,
868 0.66304426693021520000,
869 0.73218211874028970000,
870 0.79448379596794250000,
871 0.84936761373256990000,
872 0.89632115576605220000,
873 0.93490607593773970000,
874 0.96476225558750640000,
875 0.98561151154526840000,
876 0.99726386184948160000
877 };
878
879
880 private static final double[] W_32 = {
881 0.00701861000947013600,
882 0.01627439473090571200,
883 0.02539206530926214200,
884 0.03427386291302141000,
885 0.04283589802222658600,
886 0.05099805926237621600,
887 0.05868409347853559000,
888 0.06582222277636193000,
889 0.07234579410884862000,
890 0.07819389578707042000,
891 0.08331192422694673000,
892 0.08765209300440380000,
893 0.09117387869576390000,
894 0.09384439908080441000,
895 0.09563872007927487000,
896 0.09654008851472784000,
897 0.09654008851472784000,
898 0.09563872007927487000,
899 0.09384439908080441000,
900 0.09117387869576390000,
901 0.08765209300440380000,
902 0.08331192422694673000,
903 0.07819389578707042000,
904 0.07234579410884862000,
905 0.06582222277636193000,
906 0.05868409347853559000,
907 0.05099805926237621600,
908 0.04283589802222658600,
909 0.03427386291302141000,
910 0.02539206530926214200,
911 0.01627439473090571200,
912 0.00701861000947013600
913 };
914
915
916 private static final double[] P_40 = {
917 -0.99823770971055930000,
918 -0.99072623869945710000,
919 -0.97725994998377420000,
920 -0.95791681921379170000,
921 -0.93281280827867660000,
922 -0.90209880696887420000,
923 -0.86595950321225960000,
924 -0.82461223083331170000,
925 -0.77830565142651940000,
926 -0.72731825518992710000,
927 -0.67195668461417960000,
928 -0.61255388966798030000,
929 -0.54946712509512820000,
930 -0.48307580168617870000,
931 -0.41377920437160500000,
932 -0.34199409082575850000,
933 -0.26815218500725370000,
934 -0.19269758070137110000,
935 -0.11608407067525522000,
936 -0.03877241750605081600,
937 0.03877241750605081600,
938 0.11608407067525522000,
939 0.19269758070137110000,
940 0.26815218500725370000,
941 0.34199409082575850000,
942 0.41377920437160500000,
943 0.48307580168617870000,
944 0.54946712509512820000,
945 0.61255388966798030000,
946 0.67195668461417960000,
947 0.72731825518992710000,
948 0.77830565142651940000,
949 0.82461223083331170000,
950 0.86595950321225960000,
951 0.90209880696887420000,
952 0.93281280827867660000,
953 0.95791681921379170000,
954 0.97725994998377420000,
955 0.99072623869945710000,
956 0.99823770971055930000
957 };
958
959
960 private static final double[] W_40 = {
961 0.00452127709853309800,
962 0.01049828453115270400,
963 0.01642105838190797300,
964 0.02224584919416689000,
965 0.02793700698002338000,
966 0.03346019528254786500,
967 0.03878216797447199000,
968 0.04387090818567333000,
969 0.04869580763507221000,
970 0.05322784698393679000,
971 0.05743976909939157000,
972 0.06130624249292891000,
973 0.06480401345660108000,
974 0.06791204581523394000,
975 0.07061164739128681000,
976 0.07288658239580408000,
977 0.07472316905796833000,
978 0.07611036190062619000,
979 0.07703981816424793000,
980 0.07750594797842482000,
981 0.07750594797842482000,
982 0.07703981816424793000,
983 0.07611036190062619000,
984 0.07472316905796833000,
985 0.07288658239580408000,
986 0.07061164739128681000,
987 0.06791204581523394000,
988 0.06480401345660108000,
989 0.06130624249292891000,
990 0.05743976909939157000,
991 0.05322784698393679000,
992 0.04869580763507221000,
993 0.04387090818567333000,
994 0.03878216797447199000,
995 0.03346019528254786500,
996 0.02793700698002338000,
997 0.02224584919416689000,
998 0.01642105838190797300,
999 0.01049828453115270400,
1000 0.00452127709853309800
1001 };
1002
1003
1004 private static final double[] P_48 = {
1005 -0.99877100725242610000,
1006 -0.99353017226635080000,
1007 -0.98412458372282700000,
1008 -0.97059159254624720000,
1009 -0.95298770316043080000,
1010 -0.93138669070655440000,
1011 -0.90587913671556960000,
1012 -0.87657202027424800000,
1013 -0.84358826162439350000,
1014 -0.80706620402944250000,
1015 -0.76715903251574020000,
1016 -0.72403413092381470000,
1017 -0.67787237963266400000,
1018 -0.62886739677651370000,
1019 -0.57722472608397270000,
1020 -0.52316097472223300000,
1021 -0.46690290475095840000,
1022 -0.40868648199071680000,
1023 -0.34875588629216070000,
1024 -0.28736248735545555000,
1025 -0.22476379039468908000,
1026 -0.16122235606889174000,
1027 -0.09700469920946270000,
1028 -0.03238017096286937000,
1029 0.03238017096286937000,
1030 0.09700469920946270000,
1031 0.16122235606889174000,
1032 0.22476379039468908000,
1033 0.28736248735545555000,
1034 0.34875588629216070000,
1035 0.40868648199071680000,
1036 0.46690290475095840000,
1037 0.52316097472223300000,
1038 0.57722472608397270000,
1039 0.62886739677651370000,
1040 0.67787237963266400000,
1041 0.72403413092381470000,
1042 0.76715903251574020000,
1043 0.80706620402944250000,
1044 0.84358826162439350000,
1045 0.87657202027424800000,
1046 0.90587913671556960000,
1047 0.93138669070655440000,
1048 0.95298770316043080000,
1049 0.97059159254624720000,
1050 0.98412458372282700000,
1051 0.99353017226635080000,
1052 0.99877100725242610000
1053 };
1054
1055
1056 private static final double[] W_48 = {
1057 0.00315334605230596250,
1058 0.00732755390127620800,
1059 0.01147723457923446900,
1060 0.01557931572294386600,
1061 0.01961616045735556700,
1062 0.02357076083932435600,
1063 0.02742650970835688000,
1064 0.03116722783279807000,
1065 0.03477722256477045000,
1066 0.03824135106583080600,
1067 0.04154508294346483000,
1068 0.04467456085669424000,
1069 0.04761665849249054000,
1070 0.05035903555385448000,
1071 0.05289018948519365000,
1072 0.05519950369998416500,
1073 0.05727729210040315000,
1074 0.05911483969839566000,
1075 0.06070443916589384000,
1076 0.06203942315989268000,
1077 0.06311419228625403000,
1078 0.06392423858464817000,
1079 0.06446616443595010000,
1080 0.06473769681268386000,
1081 0.06473769681268386000,
1082 0.06446616443595010000,
1083 0.06392423858464817000,
1084 0.06311419228625403000,
1085 0.06203942315989268000,
1086 0.06070443916589384000,
1087 0.05911483969839566000,
1088 0.05727729210040315000,
1089 0.05519950369998416500,
1090 0.05289018948519365000,
1091 0.05035903555385448000,
1092 0.04761665849249054000,
1093 0.04467456085669424000,
1094 0.04154508294346483000,
1095 0.03824135106583080600,
1096 0.03477722256477045000,
1097 0.03116722783279807000,
1098 0.02742650970835688000,
1099 0.02357076083932435600,
1100 0.01961616045735556700,
1101 0.01557931572294386600,
1102 0.01147723457923446900,
1103 0.00732755390127620800,
1104 0.00315334605230596250
1105 };
1106
1107
1108
1109 private final double[] nodePoints;
1110
1111
1112 private final double[] nodeWeights;
1113
1114
1115
1116
1117
1118 GaussQuadrature(final int numberOfPoints) {
1119
1120 switch(numberOfPoints) {
1121 case 12 :
1122 this.nodePoints = P_12.clone();
1123 this.nodeWeights = W_12.clone();
1124 break;
1125 case 16 :
1126 this.nodePoints = P_16.clone();
1127 this.nodeWeights = W_16.clone();
1128 break;
1129 case 20 :
1130 this.nodePoints = P_20.clone();
1131 this.nodeWeights = W_20.clone();
1132 break;
1133 case 24 :
1134 this.nodePoints = P_24.clone();
1135 this.nodeWeights = W_24.clone();
1136 break;
1137 case 32 :
1138 this.nodePoints = P_32.clone();
1139 this.nodeWeights = W_32.clone();
1140 break;
1141 case 40 :
1142 this.nodePoints = P_40.clone();
1143 this.nodeWeights = W_40.clone();
1144 break;
1145 case 48 :
1146 default :
1147 this.nodePoints = P_48.clone();
1148 this.nodeWeights = W_48.clone();
1149 break;
1150 }
1151
1152 }
1153
1154
1155
1156
1157
1158
1159
1160
1161 public double[] integrate(final UnivariateVectorFunction f,
1162 final double lowerBound, final double upperBound) {
1163
1164 final double[] adaptedPoints = nodePoints.clone();
1165 final double[] adaptedWeights = nodeWeights.clone();
1166 transform(adaptedPoints, adaptedWeights, lowerBound, upperBound);
1167 return basicIntegrate(f, adaptedPoints, adaptedWeights);
1168 }
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181 private void transform(final double[] points, final double[] weights,
1182 final double a, final double b) {
1183
1184 final double scale = (b - a) / 2;
1185 final double shift = a + scale;
1186 for (int i = 0; i < points.length; i++) {
1187 points[i] = points[i] * scale + shift;
1188 weights[i] *= scale;
1189 }
1190 }
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201 private double[] basicIntegrate(final UnivariateVectorFunction f,
1202 final double[] points,
1203 final double[] weights) {
1204 double x = points[0];
1205 double w = weights[0];
1206 double[] v = f.value(x);
1207 final double[] y = new double[v.length];
1208 for (int j = 0; j < v.length; j++) {
1209 y[j] = w * v[j];
1210 }
1211 final double[] t = y.clone();
1212 final double[] c = new double[v.length];
1213 final double[] s = t.clone();
1214 for (int i = 1; i < points.length; i++) {
1215 x = points[i];
1216 w = weights[i];
1217 v = f.value(x);
1218 for (int j = 0; j < v.length; j++) {
1219 y[j] = w * v[j] - c[j];
1220 t[j] = s[j] + y[j];
1221 c[j] = (t[j] - s[j]) - y[j];
1222 s[j] = t[j];
1223 }
1224 }
1225 return s;
1226 }
1227
1228 }
1229
1230
1231
1232
1233
1234
1235
1236
1237 private class FourierCjSjCoefficients {
1238
1239
1240 private final int jMax;
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253 private final double[][] cCoef;
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266 private final double[][] sCoef;
1267
1268
1269
1270
1271
1272
1273 FourierCjSjCoefficients(final SpacecraftState state, final int jMax)
1274 throws OrekitException {
1275
1276 this.jMax = jMax;
1277
1278
1279 final int rows = jMax + 1;
1280 cCoef = new double[rows][6];
1281 sCoef = new double[rows][6];
1282
1283
1284 computeCoefficients(state);
1285 }
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296 private void computeCoefficients(final SpacecraftState state)
1297 throws OrekitException {
1298
1299 final double[] ll = getLLimits(state);
1300
1301 if (ll[0] < ll[1]) {
1302
1303 final double ooPI = 1 / FastMath.PI;
1304
1305
1306 for (int j = 0; j <= jMax; j++) {
1307 final double[] curentCoefficients =
1308 integrator.integrate(new IntegrableFunction(state, false, j), ll[0], ll[1]);
1309
1310
1311 for (int i = 0; i < 6; i++) {
1312 cCoef[j][i] = ooPI * curentCoefficients[i];
1313 sCoef[j][i] = ooPI * curentCoefficients[i + 6];
1314 }
1315 }
1316 }
1317 }
1318
1319
1320
1321
1322
1323
1324 public double getCij(final int i, final int j) {
1325 return cCoef[j][i];
1326 }
1327
1328
1329
1330
1331
1332
1333 public double getSij(final int i, final int j) {
1334 return sCoef[j][i];
1335 }
1336 }
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348 private static class GaussianShortPeriodicCoefficients implements ShortPeriodTerms {
1349
1350
1351 private static final long serialVersionUID = 20151118L;
1352
1353
1354 private final int jMax;
1355
1356
1357 private final int interpolationPoints;
1358
1359
1360 private final String coefficientsKeyPrefix;
1361
1362
1363 private final transient TimeSpanMap<Slot> slots;
1364
1365
1366
1367
1368
1369
1370
1371 GaussianShortPeriodicCoefficients(final String coefficientsKeyPrefix,
1372 final int jMax, final int interpolationPoints,
1373 final TimeSpanMap<Slot> slots) {
1374
1375 this.jMax = jMax;
1376 this.interpolationPoints = interpolationPoints;
1377 this.coefficientsKeyPrefix = coefficientsKeyPrefix;
1378 this.slots = slots;
1379 }
1380
1381
1382
1383
1384
1385 public Slot createSlot(final SpacecraftState ... meanStates) {
1386 final Slot slot = new Slot(jMax, interpolationPoints);
1387 final AbsoluteDate first = meanStates[0].getDate();
1388 final AbsoluteDate last = meanStates[meanStates.length - 1].getDate();
1389 if (first.compareTo(last) <= 0) {
1390 slots.addValidAfter(slot, first);
1391 } else {
1392 slots.addValidBefore(slot, first);
1393 }
1394 return slot;
1395 }
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407 private void computeCoefficients(final SpacecraftState state, final Slot slot,
1408 final FourierCjSjCoefficients fourierCjSj,
1409 final UijVijCoefficients uijvij,
1410 final double n, final double a)
1411 throws OrekitException {
1412
1413
1414 final AbsoluteDate date = state.getDate();
1415
1416
1417 final double k20 = computeK20(jMax, uijvij.currentRhoSigmaj);
1418
1419
1420 final double oon = 1. / n;
1421
1422 final double to2an = 1.5 * oon / a;
1423
1424 final double to4an = to2an / 2;
1425
1426
1427 final int size = jMax + 1;
1428 final double[] di1 = new double[6];
1429 final double[] di2 = new double[6];
1430 final double[][] currentCij = new double[size][6];
1431 final double[][] currentSij = new double[size][6];
1432 for (int i = 0; i < 6; i++) {
1433
1434
1435 di1[i] = -oon * fourierCjSj.getCij(i, 0);
1436 if (i == 5) {
1437 di1[i] += to2an * uijvij.getU1(0, 0);
1438 }
1439 di2[i] = 0.;
1440 if (i == 5) {
1441 di2[i] += -to4an * fourierCjSj.getCij(0, 0);
1442 }
1443
1444
1445 currentCij[0][i] = -di2[i] * k20;
1446
1447 for (int j = 1; j <= jMax; j++) {
1448
1449 currentCij[j][i] = oon * uijvij.getU1(j, i);
1450 if (i == 5) {
1451 currentCij[j][i] += -to2an * uijvij.getU2(j);
1452 }
1453 currentSij[j][i] = oon * uijvij.getV1(j, i);
1454 if (i == 5) {
1455 currentSij[j][i] += -to2an * uijvij.getV2(j);
1456 }
1457
1458
1459 currentCij[0][i] += -(currentCij[j][i] * uijvij.currentRhoSigmaj[0][j] + currentSij[j][i] * uijvij.currentRhoSigmaj[1][j]);
1460 }
1461
1462 }
1463
1464
1465 slot.cij[0].addGridPoint(date, currentCij[0]);
1466 slot.dij[1].addGridPoint(date, di1);
1467 slot.dij[2].addGridPoint(date, di2);
1468 for (int j = 1; j <= jMax; j++) {
1469 slot.cij[j].addGridPoint(date, currentCij[j]);
1470 slot.sij[j].addGridPoint(date, currentSij[j]);
1471 }
1472
1473 }
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485 private double computeK20(final int kMax, final double[][] currentRhoSigmaj) {
1486 double k20 = 0.;
1487
1488 for (int kIndex = 1; kIndex <= kMax; kIndex++) {
1489
1490
1491 double currentTerm = currentRhoSigmaj[1][kIndex] * currentRhoSigmaj[1][kIndex] +
1492 currentRhoSigmaj[0][kIndex] * currentRhoSigmaj[0][kIndex];
1493
1494
1495 currentTerm *= 2. / (kIndex * kIndex);
1496
1497
1498 k20 += currentTerm;
1499 }
1500
1501 return k20;
1502 }
1503
1504
1505 @Override
1506 public double[] value(final Orbit meanOrbit) {
1507
1508
1509 final Slot slot = slots.get(meanOrbit.getDate());
1510
1511
1512 final double L = meanOrbit.getLv();
1513
1514
1515 final double center = L - meanOrbit.getLM();
1516
1517 final double center2 = center * center;
1518
1519
1520 final double[] shortPeriodicVariation = slot.cij[0].value(meanOrbit.getDate());
1521 final double[] d1 = slot.dij[1].value(meanOrbit.getDate());
1522 final double[] d2 = slot.dij[2].value(meanOrbit.getDate());
1523 for (int i = 0; i < 6; i++) {
1524 shortPeriodicVariation[i] += center * d1[i] + center2 * d2[i];
1525 }
1526
1527 for (int j = 1; j <= JMAX; j++) {
1528 final double[] c = slot.cij[j].value(meanOrbit.getDate());
1529 final double[] s = slot.sij[j].value(meanOrbit.getDate());
1530 final double cos = FastMath.cos(j * L);
1531 final double sin = FastMath.sin(j * L);
1532 for (int i = 0; i < 6; i++) {
1533
1534 shortPeriodicVariation[i] += c[i] * cos;
1535 shortPeriodicVariation[i] += s[i] * sin;
1536 }
1537 }
1538
1539 return shortPeriodicVariation;
1540
1541 }
1542
1543
1544 public String getCoefficientsKeyPrefix() {
1545 return coefficientsKeyPrefix;
1546 }
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557 @Override
1558 public Map<String, double[]> getCoefficients(final AbsoluteDate date, final Set<String> selected)
1559 throws OrekitException {
1560
1561
1562 final Slot slot = slots.get(date);
1563
1564 final Map<String, double[]> coefficients = new HashMap<String, double[]>(2 * JMAX + 3);
1565 storeIfSelected(coefficients, selected, slot.cij[0].value(date), "d", 0);
1566 storeIfSelected(coefficients, selected, slot.dij[1].value(date), "d", 1);
1567 storeIfSelected(coefficients, selected, slot.dij[2].value(date), "d", 2);
1568 for (int j = 1; j <= JMAX; j++) {
1569 storeIfSelected(coefficients, selected, slot.cij[j].value(date), "c", j);
1570 storeIfSelected(coefficients, selected, slot.sij[j].value(date), "s", j);
1571 }
1572
1573 return coefficients;
1574
1575 }
1576
1577
1578
1579
1580
1581
1582
1583
1584
1585 private void storeIfSelected(final Map<String, double[]> map, final Set<String> selected,
1586 final double[] value, final String id, final int ... indices) {
1587 final StringBuilder keyBuilder = new StringBuilder(getCoefficientsKeyPrefix());
1588 keyBuilder.append(id);
1589 for (int index : indices) {
1590 keyBuilder.append('[').append(index).append(']');
1591 }
1592 final String key = keyBuilder.toString();
1593 if (selected.isEmpty() || selected.contains(key)) {
1594 map.put(key, value);
1595 }
1596 }
1597
1598
1599
1600
1601
1602 private Object writeReplace() throws NotSerializableException {
1603
1604
1605 final SortedSet<TimeSpanMap.Transition<Slot>> transitions = slots.getTransitions();
1606 final AbsoluteDate[] transitionDates = new AbsoluteDate[transitions.size()];
1607 final Slot[] allSlots = new Slot[transitions.size() + 1];
1608 int i = 0;
1609 for (final TimeSpanMap.Transition<Slot> transition : transitions) {
1610 if (i == 0) {
1611
1612 allSlots[i] = transition.getBefore();
1613 }
1614 if (i < transitionDates.length) {
1615 transitionDates[i] = transition.getDate();
1616 allSlots[++i] = transition.getAfter();
1617 }
1618 }
1619
1620 return new DataTransferObject(jMax, interpolationPoints, coefficientsKeyPrefix,
1621 transitionDates, allSlots);
1622
1623 }
1624
1625
1626
1627 private static class DataTransferObject implements Serializable {
1628
1629
1630 private static final long serialVersionUID = 20160319L;
1631
1632
1633 private final int jMax;
1634
1635
1636 private final int interpolationPoints;
1637
1638
1639 private final String coefficientsKeyPrefix;
1640
1641
1642 private final AbsoluteDate[] transitionDates;
1643
1644
1645 private final Slot[] allSlots;
1646
1647
1648
1649
1650
1651
1652
1653
1654 DataTransferObject(final int jMax, final int interpolationPoints,
1655 final String coefficientsKeyPrefix,
1656 final AbsoluteDate[] transitionDates, final Slot[] allSlots) {
1657 this.jMax = jMax;
1658 this.interpolationPoints = interpolationPoints;
1659 this.coefficientsKeyPrefix = coefficientsKeyPrefix;
1660 this.transitionDates = transitionDates;
1661 this.allSlots = allSlots;
1662 }
1663
1664
1665
1666
1667 private Object readResolve() {
1668
1669 final TimeSpanMap<Slot> slots = new TimeSpanMap<Slot>(allSlots[0]);
1670 for (int i = 0; i < transitionDates.length; ++i) {
1671 slots.addValidAfter(allSlots[i + 1], transitionDates[i]);
1672 }
1673
1674 return new GaussianShortPeriodicCoefficients(coefficientsKeyPrefix, jMax, interpolationPoints, slots);
1675
1676 }
1677
1678 }
1679
1680 }
1681
1682
1683
1684
1685
1686
1687
1688
1689
1690
1691
1692 private static class UijVijCoefficients {
1693
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703 private final double[][] u1ij;
1704
1705
1706
1707
1708
1709
1710
1711
1712
1713 private final double[][] v1ij;
1714
1715
1716
1717
1718
1719
1720 private final double[] u2ij;
1721
1722
1723
1724
1725
1726
1727 private final double[] v2ij;
1728
1729
1730 private final double[][] currentRhoSigmaj;
1731
1732
1733 private final FourierCjSjCoefficients fourierCjSj;
1734
1735
1736 private final int jMax;
1737
1738
1739
1740
1741
1742
1743 UijVijCoefficients(final double[][] currentRhoSigmaj, final FourierCjSjCoefficients fourierCjSj, final int jMax) {
1744 this.currentRhoSigmaj = currentRhoSigmaj;
1745 this.fourierCjSj = fourierCjSj;
1746 this.jMax = jMax;
1747
1748
1749 this.u1ij = new double[6][2 * jMax + 1];
1750 this.v1ij = new double[6][2 * jMax + 1];
1751 this.u2ij = new double[jMax + 1];
1752 this.v2ij = new double[jMax + 1];
1753
1754
1755 computeU1V1Coefficients();
1756 computeU2V2Coefficients();
1757 }
1758
1759
1760 private void computeU1V1Coefficients() {
1761
1762
1763
1764 u1ij[0][0] = 0;
1765 for (int j = 1; j <= jMax; j++) {
1766
1767 final double ooj = 1. / j;
1768
1769 for (int i = 0; i < 6; i++) {
1770
1771 u1ij[i][j] = fourierCjSj.getSij(i, j);
1772 v1ij[i][j] = fourierCjSj.getCij(i, j);
1773
1774
1775 if (j > 1) {
1776
1777 for (int kIndex = 1; kIndex <= j - 1; kIndex++) {
1778
1779
1780 u1ij[i][j] += fourierCjSj.getCij(i, j - kIndex) * currentRhoSigmaj[1][kIndex] +
1781 fourierCjSj.getSij(i, j - kIndex) * currentRhoSigmaj[0][kIndex];
1782
1783
1784
1785 v1ij[i][j] += fourierCjSj.getCij(i, j - kIndex) * currentRhoSigmaj[0][kIndex] -
1786 fourierCjSj.getSij(i, j - kIndex) * currentRhoSigmaj[1][kIndex];
1787 }
1788 }
1789
1790
1791
1792 if (j != jMax) {
1793 for (int kIndex = 1; kIndex <= jMax - j; kIndex++) {
1794
1795
1796 u1ij[i][j] += -fourierCjSj.getCij(i, j + kIndex) * currentRhoSigmaj[1][kIndex] +
1797 fourierCjSj.getSij(i, j + kIndex) * currentRhoSigmaj[0][kIndex];
1798
1799
1800
1801 v1ij[i][j] += fourierCjSj.getCij(i, j + kIndex) * currentRhoSigmaj[0][kIndex] +
1802 fourierCjSj.getSij(i, j + kIndex) * currentRhoSigmaj[1][kIndex];
1803 }
1804 }
1805
1806 for (int kIndex = 1; kIndex <= jMax; kIndex++) {
1807
1808
1809 u1ij[i][j] += -fourierCjSj.getCij(i, kIndex) * currentRhoSigmaj[1][j + kIndex] -
1810 fourierCjSj.getSij(i, kIndex) * currentRhoSigmaj[0][j + kIndex];
1811
1812
1813
1814 v1ij[i][j] += fourierCjSj.getCij(i, kIndex) * currentRhoSigmaj[0][j + kIndex] +
1815 fourierCjSj.getSij(i, kIndex) * currentRhoSigmaj[1][j + kIndex];
1816 }
1817
1818
1819 u1ij[i][j] *= -ooj;
1820 v1ij[i][j] *= ooj;
1821
1822
1823 if (i == 0) {
1824
1825 u1ij[0][0] += -u1ij[0][j] * currentRhoSigmaj[0][j] - v1ij[0][j] * currentRhoSigmaj[1][j];
1826 }
1827 }
1828 }
1829
1830
1831
1832
1833 for (int j = jMax + 1; j <= 2 * jMax; j++) {
1834
1835 final double ooj = 1. / j;
1836
1837 u1ij[0][j] = 0.;
1838 v1ij[0][j] = 0.;
1839
1840
1841 for (int kIndex = j - jMax; kIndex <= j - 1; kIndex++) {
1842
1843
1844 u1ij[0][j] += fourierCjSj.getCij(0, j - kIndex) * currentRhoSigmaj[1][kIndex] +
1845 fourierCjSj.getSij(0, j - kIndex) * currentRhoSigmaj[0][kIndex];
1846
1847
1848
1849 v1ij[0][j] += fourierCjSj.getCij(0, j - kIndex) * currentRhoSigmaj[0][kIndex] -
1850 fourierCjSj.getSij(0, j - kIndex) * currentRhoSigmaj[1][kIndex];
1851 }
1852 for (int kIndex = 1; kIndex <= jMax; kIndex++) {
1853
1854
1855 u1ij[0][j] += -fourierCjSj.getCij(0, kIndex) * currentRhoSigmaj[1][j + kIndex] -
1856 fourierCjSj.getSij(0, kIndex) * currentRhoSigmaj[0][j + kIndex];
1857
1858
1859
1860 v1ij[0][j] += fourierCjSj.getCij(0, kIndex) * currentRhoSigmaj[0][j + kIndex] +
1861 fourierCjSj.getSij(0, kIndex) * currentRhoSigmaj[1][j + kIndex];
1862 }
1863
1864
1865 u1ij[0][j] *= -ooj;
1866 v1ij[0][j] *= ooj;
1867 }
1868 }
1869
1870
1871
1872
1873
1874
1875 private void computeU2V2Coefficients() {
1876 for (int j = 1; j <= jMax; j++) {
1877
1878 final double ooj = 1. / j;
1879
1880
1881 u2ij[j] = v1ij[0][j];
1882 v2ij[j] = u1ij[0][j];
1883
1884
1885 if (j > 1) {
1886 for (int l = 1; l <= j - 1; l++) {
1887
1888
1889 u2ij[j] += u1ij[0][j - l] * currentRhoSigmaj[1][l] +
1890 v1ij[0][j - l] * currentRhoSigmaj[0][l];
1891
1892
1893
1894 v2ij[j] += u1ij[0][j - l] * currentRhoSigmaj[0][l] -
1895 v1ij[0][j - l] * currentRhoSigmaj[1][l];
1896 }
1897 }
1898
1899 for (int l = 1; l <= jMax; l++) {
1900
1901
1902
1903
1904 u2ij[j] += -u1ij[0][j + l] * currentRhoSigmaj[1][l] +
1905 u1ij[0][l] * currentRhoSigmaj[1][j + l] +
1906 v1ij[0][j + l] * currentRhoSigmaj[0][l] -
1907 v1ij[0][l] * currentRhoSigmaj[0][j + l];
1908
1909
1910
1911
1912
1913 u2ij[j] += u1ij[0][j + l] * currentRhoSigmaj[0][l] +
1914 u1ij[0][l] * currentRhoSigmaj[0][j + l] +
1915 v1ij[0][j + l] * currentRhoSigmaj[1][l] +
1916 v1ij[0][l] * currentRhoSigmaj[1][j + l];
1917 }
1918
1919
1920 u2ij[j] *= -ooj;
1921 v2ij[j] *= ooj;
1922 }
1923 }
1924
1925
1926
1927
1928
1929
1930
1931 public double getU1(final int j, final int i) {
1932 return u1ij[i][j];
1933 }
1934
1935
1936
1937
1938
1939
1940
1941 public double getV1(final int j, final int i) {
1942 return v1ij[i][j];
1943 }
1944
1945
1946
1947
1948
1949
1950 public double getU2(final int j) {
1951 return u2ij[j];
1952 }
1953
1954
1955
1956
1957
1958
1959 public double getV2(final int j) {
1960 return v2ij[j];
1961 }
1962 }
1963
1964
1965 private static class Slot implements Serializable {
1966
1967
1968 private static final long serialVersionUID = 20160319L;
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982 private final ShortPeriodicsInterpolatedCoefficient[] dij;
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996 private final ShortPeriodicsInterpolatedCoefficient[] cij;
1997
1998
1999
2000
2001
2002
2003
2004
2005
2006
2007
2008
2009
2010 private final ShortPeriodicsInterpolatedCoefficient[] sij;
2011
2012
2013
2014
2015
2016 Slot(final int jMax, final int interpolationPoints) {
2017
2018 dij = new ShortPeriodicsInterpolatedCoefficient[3];
2019 cij = new ShortPeriodicsInterpolatedCoefficient[jMax + 1];
2020 sij = new ShortPeriodicsInterpolatedCoefficient[jMax + 1];
2021
2022
2023 for (int j = 0; j <= jMax; j++) {
2024 cij[j] = new ShortPeriodicsInterpolatedCoefficient(interpolationPoints);
2025 if (j > 0) {
2026 sij[j] = new ShortPeriodicsInterpolatedCoefficient(interpolationPoints);
2027 }
2028
2029 if (j == 1 || j == 2) {
2030 dij[j] = new ShortPeriodicsInterpolatedCoefficient(interpolationPoints);
2031 }
2032 }
2033
2034 }
2035
2036 }
2037
2038 }